Custom Commands *************** Frequently the build process for a software project goes beyond simply compiling libraries and executables. In many cases, additional tasks may be required during or after the build process. Common examples include: compiling documentation using a documentation package; generating source files by running another executable; generating files using tools for which CMake doesn't have rules (such as lex and yacc); moving the resulting executables; post processing the executable, etc. CMake supports these additional tasks using the :command:`add_custom_command` and :command:`add_custom_target` commands. This chapter will describe how to use custom commands and targets to perform complex tasks that CMake does not inherently support. .. index:: single: portability issues single: shell commands single: batch commands Portable Custom Commands ======================== Before going into detail on how to use custom commands, we will discuss how to deal with some of their portability issues. Custom commands typically involve running programs with files as inputs or outputs. Even a simple command, such as copying a file, can be tricky to do in a cross-platform way. For example, copying a file on UNIX is done with the ``cp`` command, while on Windows it is done with the ``copy`` command. To make matters worse, frequently the names of files will change on different platforms. Executables on Windows end with .exe, while on UNIX they do not. Even between UNIX implementations there are differences, such as which extensions are used for shared libraries; .so, .sl, .dylib, etc. CMake provides three main tools for handling these differences. The first is the ``-E`` option (short for execute) to :manual:`cmake <cmake(1)>`. When the :manual:`cmake <cmake(1)>` executable is passed the ``-E`` option, it acts as a general purpose, cross-platform utility command. The arguments following the ``-E`` option indicate what :manual:`cmake <cmake(1)>` should do. These options provide a platform-independent way to perform a few common tasks including copy or remove files, compare and conditionally copy, time, create symlinks and others. The :manual:`cmake <cmake(1)>` executable can be referenced by using the :variable:`CMAKE_COMMAND` variable in your CMakeLists files, as later examples will show. Of course, CMake doesn't limit you to using ``cmake -E`` in all your custom commands. You can use any command that you like, though it's important to consider portability issues when doing it. A common practice is to use :command:`find_program` to find an executable (Perl, for example), and then use that executable in your custom commands. The second tool that CMake provides to address portability issues is a number of variables describing the characteristics of the platform. The :manual:`cmake-variables(7)` manual lists many variables that are useful for custom commands that need to reference files with platform-dependent names. These include :variable:`CMAKE_EXECUTABLE_SUFFIX`, :variable:`CMAKE_SHARED_LIBRARY_PREFIX`, etc. which describe file naming conventions. Finally, CMake supports :manual:`generator expressions <cmake-generator-expressions(7)>` in custom commands. These are expressions that use the special syntax ``$<...>`` and are not evaluated while processing CMake input files, but are instead delayed until generation of the final build system. Therefore, the values substituted for them know all the details of their evaluation context, including the current build configuration and all build properties associated with a target. .. index:: single: command ; add_custom_command single: targets ; custom commands single: Visual Studio .. _add_custom_command: Add Custom Command ================== Now we will consider the signature for :command:`add_custom_command`. In Makefile terminology, :command:`add_custom_command` adds a rule to a Makefile. For those more familiar with Visual Studio, it adds a custom build step to a file. :command:`add_custom_command` has two main signatures: one for adding a custom command to a target and one for adding a custom command to build a file. The target is the name of a CMake target (executable, library, or custom) to which you want to add the custom command. There is a choice of when the custom command should be executed. You can specify as many commands as you want for a custom command. They will be executed in the order specified. Now let us consider a simple custom command for copying an executable once it has been built. .. index:: single: command ; add_custom_target single: command ; add_executable .. code-block:: cmake # first define the executable target as usual add_executable(Foo bar.c) # then add the custom command to copy it add_custom_command( TARGET Foo POST_BUILD COMMAND ${CMAKE_COMMAND} ARGS -E copy $<TARGET_FILE:Foo> /testing_department/files ) The first command in this example is the standard command for creating an executable from a list of source files. In this case, an executable named ``Foo`` is created from the source file ``bar.c``. Next is the :command:`add_custom_command` invocation. Here the target is simply ``Foo`` and we are adding a post build command. The command to execute is :manual:`cmake <cmake(1)>` which has its full path specified in the :variable:`CMAKE_COMMAND` variable. Its arguments are ``-E copy`` and the source and destination locations. In this case, it will copy the ``Foo`` executable from where it was built into the ``/testing_department/files`` directory. Note that the ``TARGET`` parameter accepts a CMake target (``Foo`` in this example), but arguments specified to the ``COMMAND`` parameter normally require full paths. In this case, we pass to ``cmake -E copy``, the full path to the executable referenced via the ``$<TARGET_FILE:...>`` generator expression. .. index:: single: command ; add_custom_command Generate a File =============== The second use for :command:`add_custom_command` is to add a rule for how to build an output file. Here the rule provided will replace any current rules for building that file. Keep in mind that :command:`add_custom_command` output must be consumed by a target in the same scope. As discussed later, the :command:`add_custom_target` command can be used for this. .. index:: single: source files ; generated Using an Executable to Build a Source File ------------------------------------------ Sometimes a software project builds an executable that is then used for generating source files, which are used to build other executables or libraries. This may sound like an odd case, but it occurs quite frequently. One example is the build process for the TIFF library, which creates an executable that is then run to generate a source file that has system specific information in it. This file is then used as a source file in building the main TIFF library. Another example is the Visualization Toolkit (VTK), which builds an executable called ``vtkWrapTcl`` that wraps C++ classes into Tcl. The executable is built and then used to create more source files for the build process. .. code-block:: cmake ################################################### # Test using a compiled program to create a file #################################################### # add the executable that will create the file # build creator executable from creator.cxx add_executable(creator creator.cxx) # add the custom command to produce created.c add_custom_command( OUTPUT ${PROJECT_BINARY_DIR}/created.c DEPENDS creator COMMAND creator ${PROJECT_BINARY_DIR}/created.c ) # add an executable that uses created.c add_executable(Foo ${PROJECT_BINARY_DIR}/created.c) The first part of this example produces the ``creator`` executable from the source file ``creator.cxx``. The custom command then sets up a rule for producing the source file ``created.c`` by running the executable ``creator``. The custom command depends on the ``creator`` target and writes its result into the output tree (:variable:`PROJECT_BINARY_DIR`). Finally, an executable target called ``Foo`` is added, which is built using the ``created.c`` source file. CMake will create all the required rules in the Makefile (or Visual Studio workspace) so that when you build the project, the ``creator`` executable will be built, and run to create ``created.c``, which will then be used to build the ``Foo`` executable. .. index:: single: command ; add_custom_target single: custom targets single: targets ; custom Adding a Custom Target ====================== In the discussion so far, CMake targets have generally referred to executables and libraries. CMake supports a more general notion of targets, called custom targets, which can be used whenever you want the notion of a target but without the end product being a library or an executable. Examples of custom targets include targets to build documentation, run tests, or update web pages. To add a custom target, use the :command:`add_custom_target` command. The name specified will be the name given to the target. You can use that name to specifically build that target with Makefiles (make name) or Visual Studio (right-click on the target and then select Build). If the optional ``ALL`` argument is specified, this target will be included in the ``ALL_BUILD`` target and will automatically be built whenever the Makefile or Project is built. The command and arguments are optional; if specified, they will be added to the target as a post-build command. For custom targets that will only execute a command this is all you will need. More complex custom targets may depend on other files, in these cases the ``DEPENDS`` arguments are used to list which files this target depends on. We will consider examples of both cases. First, let us look at a custom target that has no dependencies: .. code-block:: cmake add_custom_target(FooJAR ALL ${JAR} -cvf "\"${PROJECT_BINARY_DIR}/Foo.jar\"" "\"${PROJECT_SOURCE_DIR}/Java\"" ) With the above definition, whenever the ``FooJAR`` target is built, it will run Java's Archiver (jar) to create the ``Foo.jar`` file from java classes in the ``${PROJECT_SOURCE_DIR}/Java`` directory. In essence, this type of custom target allows the developer to tie a command to a target so that it can be conveniently invoked during the build process. Now let us consider a more complex example that roughly models the generation of PDF files from LaTeX. In this case, the custom target depends on other generated files (mainly the end product .pdf files) .. index:: single: command ; add_custom_command single: command ; add_custom_target single: LaTeX custom target .. code-block:: cmake # Add the rule to build the .dvi file from the .tex # file. This relies on LATEX being set correctly # add_custom_command( OUTPUT ${PROJECT_BINARY_DIR}/doc1.dvi DEPENDS ${PROJECT_SOURCE_DIR}/doc1.tex COMMAND ${LATEX} ${PROJECT_SOURCE_DIR}/doc1.tex ) # Add the rule to produce the .pdf file from the .dvi # file. This relies on DVIPDF being set correctly # add_custom_command( OUTPUT ${PROJECT_BINARY_DIR}/doc1.pdf DEPENDS ${PROJECT_BINARY_DIR}/doc1.dvi COMMAND ${DVIPDF} ${PROJECT_BINARY_DIR}/doc1.dvi ) # finally add the custom target that when invoked # will cause the generation of the pdf file # add_custom_target(TDocument ALL DEPENDS ${PROJECT_BINARY_DIR}/doc1.pdf ) This example makes use of both :command:`add_custom_command` and :command:`add_custom_target`. The two :command:`add_custom_command` invocations are used to specify the rules for producing a .pdf file from a .tex file. In this case, there are two steps and two custom commands. First a .dvi file is produced from the .tex file by running LaTeX, then the .dvi file is processed to produce the desired .pdf file. Finally, a custom target is added called TDocument. Its command simply echoes out what it is doing, while the real work is done by the two custom commands. The ``DEPENDS`` argument sets up a dependency between the custom target and the custom commands. When TDocument is built, it will first look to see if all of its dependencies are built. If any are not built, it will invoke the appropriate custom commands to build them. This example can be shortened by combining the two custom commands into one custom command, as shown in the following example: .. code-block:: cmake # Add the rule to build the .pdf file from the .tex # file. This relies on LATEX and DVIPDF being set correctly # add_custom_command( OUTPUT ${PROJECT_BINARY_DIR}/doc1.pdf DEPENDS ${PROJECT_SOURCE_DIR}/doc1.tex COMMAND ${LATEX} ${PROJECT_SOURCE_DIR}/doc1.tex COMMAND ${DVIPDF} ${PROJECT_BINARY_DIR}/doc1.dvi ) # finally add the custom target that when invoked # will cause the generation of the pdf file add_custom_target(TDocument ALL DEPENDS ${PROJECT_BINARY_DIR}/doc1.pdf ) Now consider a case where the documentation consists of multiple files. The above example can be modified to handle many files by using a list of inputs and a :command:`foreach` loop. For example .. index:: single: command ; foreach single: command ; set .. code-block:: cmake # set the list of documents to process set(DOCS doc1 doc2 doc3) # add the custom commands for each document foreach(DOC ${DOCS}) add_custom_command( OUTPUT ${PROJECT_BINARY_DIR}/${DOC}.pdf DEPENDS ${PROJECT_SOURCE_DIR}/${DOC}.tex COMMAND ${LATEX} ${PROJECT_SOURCE_DIR}/${DOC}.tex COMMAND ${DVIPDF} ${PROJECT_BINARY_DIR}/${DOC}.dvi ) # build a list of all the results list(APPEND DOC_RESULTS ${PROJECT_BINARY_DIR}/${DOC}.pdf) endforeach() # finally add the custom target that when invoked # will cause the generation of the pdf file add_custom_target(TDocument ALL DEPENDS ${DOC_RESULTS} ) In this example, building the custom target ``TDocument`` will cause all of the specified .pdf files to be generated. Adding a new document to the list is simply a matter of adding its filename to the ``DOCS`` variable at the top of the example. Specifying Dependencies and Outputs =================================== When using custom commands and custom targets you will often be specifying dependencies. When you specify a dependency or the output of a custom command, you should always specify the full path. For example, if the command produces ``foo.h`` in the binary tree then its output should be something like ``${PROJECT_BINARY_DIR}/foo.h``. CMake will try to determine the correct path for the file if it is not specified; complex projects frequently end up using files in both the source and build trees, this can eventually lead to errors if the full paths are not specified. When specifying a target as a dependency, you can leave off the full path and executable extension, referencing it simply by its name. Consider the specification of the generator target as an :command:`add_custom_command` dependency in the example earlier in this chapter. CMake recognizes ``creator`` as matching an existing target and properly handles the dependencies. When There Isn't One Rule For One Output ======================================== There are a couple of unusual cases that can arise when using custom commands that warrant further explanation. The first is a case where one command (or executable) can create multiple outputs, and the second is when multiple commands can be used to create a single output. A Single Command Producing Multiple Outputs ------------------------------------------- In CMake, a custom command can produce multiple outputs simply by listing multiple outputs after the ``OUTPUT`` keyword. CMake will create the correct rules for your build system so that no matter which output is required for a target, the right rules will be run. If the executable happens to produce a few outputs but the build process is only using one of them, then you can simply ignore the other outputs when creating your custom command. Say that the executable produces a source file that is used in the build process, and also an execution log that is not used. The custom command should specify the source file as the output and ignore the fact that a log file is also generated. Another case of having one command with multiple outputs is when the command is the same but the arguments to it change. This is effectively the same as having a different command, and each case should have its own custom command. An example of this was the documentation example, where a custom command was added for each .tex file. The command is the same but the arguments passed to it change each time. Having One Output That Can Be Generated By Different Commands ------------------------------------------------------------- In rare cases, you may find that you have more than one command that you can use to generate an output. Most build systems, such as make and Visual Studio, do not support this and likewise CMake does not. There are two common approaches that can be used to resolve this. If you truly have two different commands that produce the same output and no other significant outputs, then you can simply pick one of them and create a custom command for it. In more complex cases there are multiple commands with multiple outputs; for example: :: Command1 produces foo.h and bar.h Command2 produces widget.h and bar.h There are a few approaches that can be used in this case. You might consider combining both commands and all three outputs into a single custom command, so that whenever one output is required, all three are built at the same time. You could also create three custom commands, one for each unique output. The custom command for ``foo.h`` would invoke Command1, while the one for ``widget.h`` would invoke Command2. When specifying the custom command for ``bar.h``, you could choose either Command1 or Command2.